Sami

HomeAboutBlogProjects

TypeScript vs JSDoc - Type Safety in JavaScript

The Strengths and Weaknesses

Sat Nov 02 2024

When beginning a new JavaScript project, type safety always comes up as a critical consideration. The two main contenders - TypeScript and JSDoc - each offer distinct approaches to adding types to JavaScript. Let's dive deep into their tradeoffs and see how they actually work in production.

TypeScript: Not Just Types

TypeScript's appeal goes beyond its type system. While "TypeScript is a typed superset of JavaScript" is the common tagline, its powerful features make it a compelling choice for complex applications:

// TypeScript's type inference is surprisingly smart
const nums = [1, 2, 3]; // Type: number[]
const first = nums[0]; // Type: number

// Structural typing allows for flexible interfaces
interface User {
  id: number;
  name: string;
  email?: string; // Optional properties
}

// TypeScript understands this satisfies User
const user = {
  id: 1,
  name: "Sam",
  website: "https://example.com", // Extra properties are fine
};

// Union types enable precise modeling
type Status = "draft" | "published" | "archived";

Where TypeScript Shines

  1. Project-Wide Refactoring

    • Renaming properties across your codebase
    • Finding all usages of a function/type
    • Catching breaking changes early
  2. Rich IDE Integration

    • Instant feedback on type errors
    • Precise autocomplete suggestions
    • Quick fixes and code actions
  3. Type Inference

    • Often needs minimal type annotations
    • Understands complex patterns
    • Helps catch subtle bugs

JSDoc: The Lightweight Alternative

JSDoc takes a different approach - it adds type information through comments, making it a zero-build-step solution:

/**
 * @typedef {Object} User
 * @property {number} id
 * @property {string} name
 * @property {string} [email]
 */

/**
 * Creates a new user
 * @param {string} name
 * @param {string} [email]
 * @returns {User}
 */
function createUser(name, email) {
  return {
    id: Date.now(),
    name,
    ...(email && { email }),
  };
}

The JSDoc Advantage

  1. No Build Step

    • Works with plain JavaScript
    • Quick to add to existing projects
    • Zero configuration needed
  2. Gradual Adoption

    • Add types where they matter most
    • Mix typed and untyped code freely
    • Great for legacy codebases
  3. Built-in Documentation

    • Types serve as documentation
    • IDEs show type info on hover
    • Generates documentation websites

Real-World Considerations

The choice between TypeScript and JSDoc often comes down to practical factors:

When to Choose TypeScript

  • Large teams working on complex domains
  • Projects that need strict type enforcement
  • When you want advanced language features
  • If you're using other typed languages

When JSDoc Makes Sense

  • Adding types to existing JavaScript
  • Small to medium-sized projects
  • When build complexity is a concern
  • If you need to support non-TypeScript tools

A Hybrid Approach

You can actually leverage both - TypeScript understands JSDoc annotations perfectly. This means you can:

  • Start with JSDoc annotations in your JavaScript
  • Gradually move to .ts files where needed
  • Keep JSDoc for documentation while using TypeScript for type-checking

Here's what that looks like:

// Using JSDoc with TypeScript
/** @type {Array<number>} */
const numbers = [1, 2, 3];

/**
 * A user in our system
 * @typedef {Object} User
 */
interface User {
  id: number;
  name: string;
}

The Performance Impact

One often overlooked aspect is the impact on development and build performance:

TypeScript

  • Initial setup requires build configuration
  • Incremental builds are very fast
  • Type checking can be moved to a separate process
  • IDE performance stays snappy with large codebases

JSDoc

  • No build step impact
  • IDE has to parse comments (can be slower)
  • Type checking through // @ts-check is per-file
  • Documentation generation is a separate step

Advanced Type Safety Examples

Let's look at some more complex patterns in both systems:

// TypeScript's advanced type features
type JSONValue =
  | string
  | number
  | boolean
  | null
  | JSONValue[]
  | { [key: string]: JSONValue };

// Utility types
type Partial<T> = { [P in keyof T]?: T[P] };
type Required<T> = { [P in keyof T]-?: T[P] };

// Template literal types
type Color = "red" | "blue";
type Size = "small" | "large";
type Product = `${Color}-${Size}-item`; // "red-small-item" | "red-large-item" | ...

And the JSDoc equivalent:

/**
 * @typedef {string|number|boolean|null|Array<JSONValue>|Object<string, JSONValue>} JSONValue
 */

/**
 * @template T
 * @typedef {Object} PartialType
 * @property {T[keyof T]} [key]
 */

/**
 * @typedef {"red"|"blue"} Color
 * @typedef {"small"|"large"} Size
 * @typedef {`${Color}-${Size}-item`} Product
 */

Conclusion

Both TypeScript and JSDoc are battle-tested solutions for adding type safety to JavaScript. TypeScript offers a more comprehensive solution with advanced language features, while JSDoc provides a lightweight approach that's perfect for gradual adoption.

The key is understanding your project's needs:

  • If you need robust tooling and strict type enforcement, go with TypeScript
  • If you want a gradual approach with zero build step, JSDoc is your friend

Remember, the goal is to write maintainable code that helps your team move faster - choose the tool that best serves that purpose.